<?php
/**
 * 身份证格式校验
 * @author zhouyang(zhouyang@sdxxtop.com)
 * @time: 2021/10/16 19:04
 */

namespace app\api\help;


class IdentityCard
{
    /**
     * 校验身份证号是否合法
     * @param string $num 待校验的身份证号
     * @return bool
     */
    public static function isValid($num)
    {
        //老身份证长度15位，新身份证长度18位
        $num = (string)($num);
        $length = strlen($num);
        if ($length == 15) { //如果是15位身份证
            
            //15位身份证没有字母
            if (!is_numeric($num)) {
                return false;
            }
            // 省市县（6位）
            $areaNum = substr($num, 0, 6);
            // 出生年月（6位）
            $dateNum = substr($num, 6, 6);
            
        } else if ($length == 18) { //如果是18位身份证
            
            //基本格式校验
            if (!preg_match('/^\d{17}[0-9xX]$/', $num)) {
                return false;
            }
            // 省市县（6位）
            $areaNum = substr($num, 0, 6);
            // 出生年月日（8位）
            $dateNum = substr($num, 6, 8);
            
        } else { //假身份证
            return false;
        }
        
        //验证地区
        if (!self::isAreaCodeValid($areaNum)) {
            return false;
        }
        
        //验证日期
        if (!self::isDateValid($dateNum)) {
            return false;
        }
        
        //验证最后一位
        if (!self::isVerifyCodeValid($num)) {
            return false;
        }
        
        return true;
    }
    
    /**
     * 省市自治区校验
     * @param string $area 省、直辖市代码
     * @return bool
     */
    private static function isAreaCodeValid($area) {
        $provinceCode = substr($area, 0, 2);
        //省代码数组
        $provinceArr=[11,12,13,14,15,21,22,23,31,32,33,34,35,36,37,41,42,43,44,45,46,50,51,52,53,54,61,62,63,64,65,71, 81,82];
        if (in_array($provinceCode,$provinceArr)){
            return true;
        }else{
            return false;
        }
    }
    
    /**
     * 验证出生日期合法性
     * @param string $date 日期
     * @return bool
     */
    private static function isDateValid($date) {
        if (strlen($date) == 6) { //15位身份证号没有年份，这里拼上年份
            $date = '19'.$date;
        }
        $year  = intval(substr($date, 0, 4));
        $month = intval(substr($date, 4, 2));
        $day   = intval(substr($date, 6, 2));
        
        //日期基本格式校验
        if (!checkdate($month, $day, $year)) {
            return false;
        }
        
        //日期格式正确，但是逻辑存在问题(如:年份大于当前年)
        $currYear = date('Y');
        if ($year > $currYear) {
            return false;
        }
        return true;
    }
    
    /**
     * 验证18位身份证最后一位
     * @param string $num 待校验的身份证号
     * @return bool
     */
    private static function isVerifyCodeValid($num)
    {
        if (strlen($num) == 18) {
            $factor = [7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2];
            $tokens = ['1', '0', 'X', '9', '8', '7', '6', '5', '4', '3', '2'];
            
            $checkSum = 0;
            for ($i = 0; $i < 17; $i++) {
                $checkSum += intval($num[$i]) * $factor[$i];
            }
            
            $mod   = $checkSum % 11;
            $token = $tokens[$mod];
            
            $lastChar = strtoupper($num[17]);
            
            if ($lastChar != $token) {
                return false;
            }
        }
        return true;
    }
}